## **RISC-V Simulator Test Plan Record**

Version: 0.03 | 2/12/2022

Editors: Hector Soto Created By: Hector Soto (sotohec@pdx.edu) December 2/12/2022

# **Preface**

This document describes tests and their results. The purpose of this document is to track tests and their results until tested code is functional with no/minimal errors. Errors/bugs encountered during tests will be recorded until fixed. If a bug is fixed, it will be removed from this document.

# **Section 1 – SimMemory**

This section goes over the testing of simulated memory in our simulation. It is defined as the class SimMemory with its definition + implementation in files "SimMemory.h", "SimMemory.cpp". SimMemory uses 4-byte aligned addressing (2 LSBs = 0). So simulated load/store instructions may have to make 2 calls instead of 1 when requesting 4-byte words or 2-byte half-words.

### 1.1 – Basic Load/Store Test (test\_mm.cpp)

"test\_mm.cpp" is a test designed for SimMemory's ability to read/write to memory consistently if given a large amount of random memory requests with random amounts of contiguous memory being requested.

Runs by default have 65,536 memory request tests. A memory request test looks like the following.

By the end of a run, around 16,800,000 attempts at reading and writing to a memory location are performed. It is possible for some attempts to be on the same address though as addresses are generated semi-randomly.

**Biggest Flaw**: Since this test immediately checks a memory location after writing it. There's no guarantee the data is stored in that location upon the request of data at a different address. The code for SimMemory indicates that is not the case, but this test doesn't check that.

There was only one bug encountered during testing, and it was from GetDataPointer() not correctly calculating where in the memory hierarchy a page was due to simply not shifting masked addresses. Test now has a 100% success rate.

```
hector@hector-VirtualBox:~/Desktop/RISC-V_Simulator/test_mm$ ./executable Success/Failure: 16839061/16839061 | Success Rate: 100.00%
```

#### 1.2 – Consistent Memory Test (test cm.cpp)

"test\_cm.cpp" is a test designed for SimMemory's advanced load/store functions which can load/store a variable amount of data. More specifically, unlike GetDataPointer(), Store() and Load() can access 1-byte, 2-bytes, or 4-bytes of contiguous memory at any address. GetDataPointer() was limited to accessing 4 bytes on 4-byte aligned addresses.

This test generates one random address and one random 4-byte word. The first byte is stored at the address. The first 2 bytes are stored at the address + 1 (account for the byte written), and the entire 4-byte word is stored at the address + 3 (account for the 3-bytes written). This process then repeats on subsequent addresses from the original. So all memory locations written to should be contiguous. Then all memory locations written to are tested to see if they hold the correct data.

Runs by default have 65,536 memory request tests. A memory request test looks like the following.

```
Address = rand() % 0xFFFFFFFF; //Addresses from 0 -> 0xFFFFFFFE.
RanData = rand() % 0xFFFFFFFF;
for(uint32 t i = 0; i < TEST COUNT; ++i)</pre>
        Memory.Store(Address + 7*i, RanData, LS_BYTE); //Store LSByte.
        Memory.Store(Address + 1 + 7*i, RanData, LS_HALF); //Store first |
        Memory.Store(Address + 3 + 7*i, RanData, LS_WORD);
        ExtraData += 2;
for(uint32 t i = 0; i < TEST COUNT; ++i)</pre>
        if(Memory.Load(Address + 7*i, LS BYTE) == (RanData & 0xFF))
        {
                ++Success:
        else
        {
                printf("Address: 0x%08X Data: 0x%08X | BYTE: 0x%08lX\n", A
        if(Memory.Load(Address + 1 + 7*i, LS_HALF) == (RanData & 0xFFFF))
                ++Success;
        else
                printf("Address: 0x%08X Data: 0x%08X | HALF: 0x%08lX\n", A
        if(Memory.Load(Address + 3 + 7*i, LS_WORD) == RanData)
        {
                ++Success;
        else
                printf("Address: 0x%08X Data: 0x%08X | WORD: 0x%08lX\n", Page 2
```

**Biggest Flaw**: Since only one random 4-byte word is generated and checked for, it is possible this test could fail to catch a consistently occurring bug in the Load/Store functions.

There was a minimal amount of bugs during testing.

Test now has a 100% success rate.

hector@hector-VirtualBox:~/Desktop/RISC-V\_Simulator/test\_cm\$ ./executable Success/Failure: 196605/196605 | Success Rate: 100.00%

## Section 2 – RVSimulator

This section goes over the testing of the overall simulator. Only major tests that checked the functionality of instructions were recorded. Minor tests for things such as reading files in correctly, terminal output, command line processing, cross-platform compilation, terminal/file output, etc, were done incrementally at the time of coding.

There was no major direct instruction tests for load/store instructions as they are just wrappers for calls to SimMemory's Load()/Store() functions which were already tested.

#### 2.1 – Immediate Instructions (Immediate.s/mem)

The following test was written in RISC-V assembly (.s) and converted into a memory file using the RISC-V assembler and manually fixing any inconsistencies from assembler optimizations. This test is performed by loading the "Immediate.mem" file into the simulator and running it. This file tests all immediate functions (ADDI, SLTI, SLTIU, XORI, ORI, ANDI, SLLI, SRLI, SRAI). Whenever a test for an instruction is passed, a value of 1 is stored in memory, when a test fails, a value of 0 is stored in memory.

Here is the first portion of the test file.

```
# This test assembly will not follow normal conventions.
# This will test ADDI, SLTI, SLTIU, XORI, ORI, ANDI, SLLI, SRLI, SRAI
# If a test passes, 1 will be stored at the current memory location.
# Otherwise a 0 will be stored.
# ADDI's result will be stored at 0x0. SLTI's at 0x4, and so on.
li
        sp,0
                        # Start at memory location 0x0.
        t0,5
                        # --- | ADDI TEST |---
li
li
        t1,8
                        # Expected ADDI result.
li
        t3,0
                        # Set test result to 0 (false).
addi
        t0,t0,3
                        # ADDI 5 + 3. Result should be 8.
        t0,t1,8
                        # If ADDI 5 + 3 != 8, ADDI failed. Jump past success.
bne
li
        t3,1
                        # Set test result to 1 (true).
        t3,0(sp)
                        # Store test result at current memory location.
                        # Increment to next memory location to store next test result.
addi
        sp,sp,4
li
                        # --- | SLTI TEST |---
        t0,-83
                        # t0 = 1 if -83 < -47.
        t0,t0,-47
slti
        t0,0(sp)
                        # SLTI gives test result. No branch code needed.
SW
        sp,sp,4
                        # Increment memory location.
addi
li
        t0,457
                        # --- | SLTIU TEST |---
        t0,t0,-1
                        # (unsigned)-1 should be > 457.
sltiu
SW
        t0,0(sp)
addi
        sp,sp,4
                        # Increment memory location.
        t1,0
                        # --- | XORI TEST |---
li
li
                        # Set test result to 0.
        t3,0
        t0,-1
                        # Load 0xFFFFFFF.
li
        t0,t0,-1
xori
                        # Result should be 0.
        t0,t1,8
bne
li
        t3,1
                        # Set test result to 1.
        t3,0(sp)
                        # Store test result
SW
                                                                                    Page 4
addi
        sp,sp,4
                        # Increment memory location.
li
        t0,0
                        # --- | ORI TEST |---
```

**Biggest Flaw**: There's no randomization for this test, so it may not catch potential edge cases that would cause bugs.

This test now has a 100% success rate.

### 2.2 – Register Instructions (Register.s/mem)

This test was created and designed nearly identically to the previous test. It tests the register instructions (ADD, SUB, SLL, SLT, SLTU, XOR, SRL, SRA, OR, AND).

Here is the first portion of the test file.

```
li
                         # Start a memory location 0x0.
        sp,0
                        # --- | ADD TEST |---
li
        t0,531
        t1,364
li
li
        t2,895
li
        t3,0
                        # Set test result to 0 (false).
        t0,t0,t1
t0,t2,8
add
                        # t0 = 531 + 364 = 895
bne
                        # Branch if result is incorrect.
                        # Set test result to 1 (true).
li
        t3,1
        t3,0(sp)
sp,sp,4
                        # Store test result at current memory location.
SW
                        # Increment memory location.
addi
        t0,279
                        # --- | SUB TEST |---
li
li
        t1,128
li
        t2,151
li
        t3,0
        t0,t0,t1
                        # t0 = 279 - 128 = 151
sub
        t0,t2,8
bne
li
        t3,1
        t3,0(sp)
SW
                        # Increment memory location.
addi
        sp,sp,4
        t0,7
                        # --- | SLL TEST |---
li
li
        t1,5
        t2,224
                        # t2 = t1 << 5
li
li
        t3,0
sll
        t0,t0,t1
                        # t0 = 7 << 5
        t0,t2,8
bne
li
        t3,1
SW
        t3,0(sp)
addi
        sp,sp,4
                        # Increment memory location.
        t0,-47
                        # --- | SLT TEST |---
li
```

**Biggest Flaw**: There's no randomization for this test, so it may not catch potential edge cases that would cause bugs.

This test now has a 100% success rate.

### 2.3 – JALR Instruction (JALR.c/s/mem)

This test was created in C, converted to RISC-V, and then converted into a memory file. The test consists of three function calls buried within each other. E.g. main() would call First(), First() called Second(), and Second() called Third(). A success was considered to be the simulation successfully jumping in the expected order and ending correctly.

```
Here is the test file.
void Third()
{
    return;
}

void Second()
{
    Third();
    return;
}

void First()
{
    Second();
    return;
}

int main()
{
    First();
    return 0;
}
```

**Biggest Flaw**: There's no randomization for this test, so it may not catch potential edge cases that would cause bugs.

This test now has a 100% success rate.

### 2.4 – The GUI, Time-Stepping, & Time Reversal

The GUI was made for extra credit and was tested incrementally during its creation. There were no specific major tests for the GUI itself, it was considered working if all UI elements showed the same information that was accessible via terminal simulation or trace files.

There was no specific major test for time-stepping. Time-stepping was implemented by moving the code that processes instructions outside of the simulation loop into its own function that would be called by the GUI or terminal simulation. Thus both the GUI and terminal are guaranteed to have the same simulation results.

Time reversal was implemented for easier testing and potential extra credit. There was no specific major tests for time reversal. If the same results were achieved as a normal test when time-reversal was used, then it was successful. Time reversal as of now appears to be completely accurate code-wise and in actual usage.